热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

循环结构与零钱问题:多题型综合解析与应用

篇首语:本文由编程笔记#小编为大家整理,主要介绍了各题型归纳总结相关的知识,希望对你有一定的参考价值。 文章目录 二分两种代码模板代码1代码2 思考有重复元素的情况山峰数组 动态规划从题目

篇首语:本文由编程笔记#小编为大家整理,主要介绍了各题型归纳总结相关的知识,希望对你有一定的参考价值。



文章目录


  • 二分
    • 两种代码模板
      • 代码1
      • 代码2

    • 思考
      • 有重复元素的情况
      • 山峰数组


  • 动态规划
    • 从题目中辨识是否用DP
      • 重复子问题
        • 剑指 Offer 46. 把数字翻译成字符串
        • 91. 解码方法

      • 最优子结构
        • 322. 零钱兑换
        • 279. 完全平方数
        • 343. 整数拆分
        • 377. 组合总和 Ⅳ

      • ==无后效性【多阶段、有约束 的决策最优化问题】==
        • 不同路径
        • 打家劫舍



  • 贪心


二分

两种代码模板


代码1


  • 代码1:在循环中查找元素
    适合知道num[mid]等于什么,才能得到结果的情况

public class Solution

// 「力扣」第 704 题:二分查找
public int search(int[] nums, int target)
int len = nums.length;
int left = 0;
int right = len - 1;
// 目标元素可能存在在区间 [left, right]
//区间还剩一个元素时,继续循环
while (left <&#61; right)
// 推荐的写法是 int mid &#61; left &#43; (right - left) / 2;
int mid &#61; (left &#43; right) / 2;
if (nums[mid] &#61;&#61; target)
return mid;
else if (nums[mid] < target)
// 目标元素可能存在在区间 [mid &#43; 1, right]
left &#61; mid &#43; 1;
else
// 目标元素可能存在在区间 [left, mid - 1]
right &#61; mid - 1;


return -1;



代码2


  • 代码2&#xff08;1&#xff09;&#xff1a;在循环体中排除目标元素一定不存在的区间
    适合 不知道num[mid]等于什么 才能得到结果&#xff0c;但知道什么情况下可以缩小区间&#xff0c;的情况。
    使用 if (nums[mid]

public class Solution

// 「力扣」第 704 题&#xff1a;二分查找
public int search(int[] nums, int target)
int len &#61; nums.length;
int left &#61; 0;
int right &#61; len - 1;
// 目标元素可能存在在区间 [left, right]
//区间还剩一个元素时&#xff0c;退出循环
while (left < right)
int mid &#61; left &#43; (right - left) / 2;
//这里注意使用nums[mid]
//若使用nums[mid] > target,则mid需要上取整&#xff1b;
if (nums[mid] < target)
// 下一轮搜索区间是 [mid &#43; 1, right]
left &#61; mid &#43; 1;
else
// 下一轮搜索区间是 [left, mid]
right &#61; mid;


if (nums[left] &#61;&#61; target)
return left;

return -1;



  • 代码2&#xff08;2&#xff09;&#xff1a;在循环体中排除目标元素一定不存在的区间

使用if (nums[mid] > target) 判断&#xff1b;会出现left &#61; mid&#xff1b;mid需要上取整:int mid &#61; left &#43; (right - left &#43; 1) / 2;防止死循环

即&#xff0c;出现left &#61; mid情况&#xff0c;mid需向上取整int mid &#61; left &#43; (right - left &#43; 1) / 2;防止死循环。

public class Solution

// 「力扣」第 704 题&#xff1a;二分查找
public int search(int[] nums, int target)
int len &#61; nums.length;
int left &#61; 0;
int right &#61; len - 1;
while (left < right)
int mid &#61; left &#43; (right - left &#43; 1) / 2;
if (nums[mid] > target)
// 下一轮搜索区间是 [left, mid - 1]
right &#61; mid - 1;
else
// 下一轮搜索区间是 [mid, right]
left &#61; mid;


if (nums[left] &#61;&#61; target)
return left;

return -1;



思考

二分的思想是&#xff0c;通过num[mid]与一个target对比&#xff0c;来缩小区间&#xff0c;


  • 当给定target时&#xff0c;自然与target比较&#xff1b;
  • 当未给定target时&#xff0c;找那些和mid比较 能产生缩小区间效果的元素&#xff0c;如&#xff1a;左右边界常作为target&#xff0c;

有重复元素的情况

针对有重复的情况&#xff0c;是将下面两种**无重复情况**下的划分&#xff1a;
nums[l] <&#61; nums[mid]
nums[l] > nums[mid])
改为下面三种划分&#xff0c;将等于的情况单独提取出来&#xff0c;【适合重复情况】
nums[l] < nums[mid]
nums[l] &#61;&#61; nums[mid] //若nums[l]不是目标值&#xff0c;因为相等&#xff0c;所以可以缩小一个范围&#xff0c;即l&#43;&#43;;
nums[l] > nums[mid])

在有序数组中进行查找一个数&#xff08;二分下标&#xff09;

在整数范围内查找一个整数&#xff08;二分答案&#xff09;


山峰数组

arr[mid] 与 arr[mid &#43; 1]比较


动态规划

找题目中的约束条件&#xff0c;然后根据约束条件定义状态



  • 动态规划的用途&#xff1a;求解多阶段决策问题
    动态规划解决的是这样一类问题&#xff1a;多阶段决策问题。这里的「阶段」就是生活语言&#xff1a;解决一个问题分很多步骤&#xff0c;每一个步骤又有很多种选择&#xff0c;这一点和「回溯算法」是一样的
    通常可以把多阶段决策问题画成一张树形图

  • 动态规划与回溯算法的区别
    「动态规划」与「回溯算法」在问题问法上的区别是&#xff1a;「动态规划」问题通常只问结果&#xff0c;即只问最优值是多少&#xff0c;或者问解决方案的个数&#xff0c;而不问具体解&#xff08;具体的解决方案&#xff09;是什么&#xff1b;
    「回溯算法」问题通常让我们给出一个问题的所有解决方案&#xff0c;要求我们返回的是一个嵌套列表



能够使用动态规划解决的问题&#xff0c;一定可以使用回溯算法解决。但是我们要清楚一个事实&#xff1a;回溯算法的时间复杂度很高。在只问最优值是多少的场景下&#xff0c;没有必要记录每个阶段的每一个步骤。动态规划方法很多时候的意义在于评估算法的上限。



从题目中辨识是否用DP


重复子问题


剑指 Offer 46. 把数字翻译成字符串

求&#xff1a;计算一个数字有多少种不同的翻译方法。
只是求有多少种&#xff0c;而不是求出每种的解决方案&#xff0c;即DP。【若求所有解决方案&#xff0c;则用回溯】


91. 解码方法

求&#xff1a;请计算并返回 解码 方法的 总数
只是求有多少种&#xff0c;而不是求出每种的解决方案&#xff0c;即DP。【若求所有解决方案&#xff0c;则用回溯】


最优子结构



它们的问法都一样&#xff1a;求解一个问题的最优值是多少&#xff0c;但没有问最优值是怎么来的。以后遇到这样的问题&#xff0c;需要有一定敏感&#xff0c;可能这个问题考察的是动态规划&#xff08;还有可能考察广度优先遍历、贪心算法&#xff09;。
分析最优子结构的重要方法依然是&#xff1a;通过研究具体的例子&#xff0c;画图分析。



322. 零钱兑换


279. 完全平方数


343. 整数拆分


377. 组合总和 Ⅳ



对于 nums &#61; [1,2,3], target &#61; 4
dp[4] &#61; dp[4 - 1] &#43; dp[4 - 2] &#43; dp[4 - 3]
即&#xff0c;
dp[target ] &#61; dp[target - nums[0] ] &#43; dp[target - nums[1] ] &#43; dp[target - nums[2] ] &#43; …【target - nums[2] >&#61;0)】


class Solution
public int combinationSum4(int[] nums, int target)
int[] dp &#61; new int[target &#43; 1];
//dp[0]没实际意思&#xff0c;由dp[1] &#61; 1 &#61; dp[0]推出&#xff1b;
dp[0] &#61; 1;
for(int j &#61; 1 ; j <&#61; target ; j&#43;&#43;)
for(int i &#61; 0 ; i < nums.length ; i&#43;&#43;)
if(j >&#61; nums[i]) dp[j] &#61; dp[j] &#43; dp[j - nums[i]];


return dp[target];



无后效性【多阶段、有约束 的决策最优化问题】


不同路径


  • 只需求出路径个数&#xff0c;不需求出具体方案&#xff1b;

打家劫舍


  • 题目只问最优值&#xff0c;并没有问最优解&#xff0c;因此可以考虑使用「动态规划」
  • 约束条件&#xff1a;在不触动警报装置的情况下&#xff0c;
    即分一个房子偷或不偷两种情况&#xff1b;

贪心



推荐阅读
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • 本文详细介绍了如何解决Uploadify插件在Internet Explorer(IE)9和10版本中遇到的点击失效及JQuery运行时错误问题。通过修改相关JavaScript代码,确保上传功能在不同浏览器环境中的一致性和稳定性。 ... [详细]
  • PHP 编程疑难解析与知识点汇总
    本文详细解答了 PHP 编程中的常见问题,并提供了丰富的代码示例和解决方案,帮助开发者更好地理解和应用 PHP 知识。 ... [详细]
  • Python 异步编程:深入理解 asyncio 库(上)
    本文介绍了 Python 3.4 版本引入的标准库 asyncio,该库为异步 IO 提供了强大的支持。我们将探讨为什么需要 asyncio,以及它如何简化并发编程的复杂性,并详细介绍其核心概念和使用方法。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文介绍了如何利用JavaScript或jQuery来判断网页中的文本框是否处于焦点状态,以及如何检测鼠标是否悬停在指定的HTML元素上。 ... [详细]
  • 深入解析JVM垃圾收集器
    本文基于《深入理解Java虚拟机:JVM高级特性与最佳实践》第二版,详细探讨了JVM中不同类型的垃圾收集器及其工作原理。通过介绍各种垃圾收集器的特性和应用场景,帮助读者更好地理解和优化JVM内存管理。 ... [详细]
  • 非公版RTX 3080显卡的革新与亮点
    本文深入探讨了图形显卡的进化历程,重点介绍了非公版RTX 3080显卡的技术特点和创新设计。 ... [详细]
  • PyCharm中配置Pylint静态代码分析工具
    本文详细介绍如何在PyCharm中配置和使用Pylint,帮助开发者进行静态代码检查,确保代码符合PEP8规范,提高代码质量。 ... [详细]
author-avatar
艾米27
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有